To be able to edit code and run cells, you need to run the notebook yourself. Where would you like to run the notebook?

In the cloud (experimental)

Binder is a free, open source service that runs scientific notebooks in the cloud! It will take a while, usually 2-7 minutes to get a session.

On your computer

(Recommended if you want to store your changes.)

  1. Copy the notebook URL:
  2. Run Pluto

    (Also see: How to install Julia and Pluto)

  3. Paste URL in the Open box

Frontmatter

If you are publishing this notebook on the web, you can set the parameters below to provide HTML metadata. This is useful for search engines and social media.

Author 1

homework 8, version 1

👀 Reading hidden code
24.3 ms

Submission by: Jazzy Doe (jazz@mit.edu)

👀 Reading hidden code
11.6 ms

Bonus homework 8: Raytracing in 3D

18.S191, fall 2020

This week we will recreate the 3D raytracer from the lectures, building on what we wrote for the 2D case. Compared to the past weeks, you will notice that this homework is a little different! We have included fewer intermediate checks, it is up to you to decide how you:

  • split up the problem is smaller functions

  • check and visualize your progress

To guide you through the homework, you can follow along with this week's live coding session. Feel free to stay close to James's code, or to come up with your own solution.

With the raytracer done, we will be able to recreate an artwork by M.C. Escher, scroll down to have a sneak peek!


For MIT students: this homework is optional.

Feel free to ask questions!

👀 Reading hidden code
41.4 ms
👀 Reading hidden code
# edit the code below to set your name and kerberos ID (i.e. email without @mit.edu)

student = (name = "Jazzy Doe", kerberos_id = "jazz")

# you might need to wait until all other cells in this notebook have completed running.
# scroll around the page to see what's up
14.7 μs

Lecture

Live coding session

👀 Reading hidden code
124 μs

Let's create a package environment:

👀 Reading hidden code
256 μs
begin
import Pkg
Pkg.activate(mktempdir())
Pkg.add([
Pkg.PackageSpec(name="Plots", version="1.6-1"),
Pkg.PackageSpec(name="PlutoUI", version="0.6.8-0.6"),
Pkg.PackageSpec(name="ImageMagick"),
Pkg.PackageSpec(name="Images", version="0.23"),
])
using Plots
using PlutoUI
using LinearAlgebra
using Images
end
👀 Reading hidden code
❔
  Activating new project at `/tmp/jl_ChwFH2`
    Updating registry at `~/.julia/registries/General.toml`
   Resolving package versions...
    Updating `/tmp/jl_ChwFH2/Project.toml`
  [6218d12a] + ImageMagick v1.3.0
  [916415d5] + Images v0.23.3
  [91a5bcdd] + Plots v1.40.7
  [7f904dfe] + PlutoUI v0.6.11
    Updating `/tmp/jl_ChwFH2/Manifest.toml`
  [621f4979] + AbstractFFTs v1.5.0
  [79e6a3ab] + Adapt v4.2.0
  [13072b0f] + AxisAlgorithms v1.0.1
  [39de3d68] + AxisArrays v0.4.7
  [d1d4a3ce] + BitFlags v0.1.9
  [aafaddc9] + CatIndices v0.2.2
  [d360d2e6] + ChainRulesCore v1.25.1
  [9e997f8a] + ChangesOfVariables v0.1.9
  [944b1d66] + CodecZlib v0.7.8
  [35d6a980] + ColorSchemes v3.26.0
  [3da002f7] + ColorTypes v0.10.12
  [c3611d14] + ColorVectorSpace v0.8.7
  [5ae59095] + Colors v0.12.11
  [34da2185] + Compat v4.16.0
  [ed09eef8] + ComputationalResources v0.3.2
  [f0e56b4a] + ConcurrentUtilities v2.5.0
  [187b0558] + ConstructionBase v1.5.8
  [d38c429a] + Contour v0.6.3
  [150eb455] + CoordinateTransformations v0.6.3
  [dc8bdbbb] + CustomUnitRanges v1.0.2
  [9a962f9c] + DataAPI v1.16.0
  [864edb3b] + DataStructures v0.18.20
  [b4f34e82] + Distances v0.10.12
  [ffbed154] + DocStringExtensions v0.9.3
  [460bff9d] + ExceptionUnwrapping v0.1.11
  [c87230d0] + FFMPEG v0.4.2
  [4f61f5a4] + FFTViews v0.3.2
  [7a1cc6ca] + FFTW v1.8.1
  [5789e2e9] + FileIO v1.17.0
  [53c48c17] + FixedPointNumbers v0.8.5
  [1fa38f19] + Format v1.3.7
  [28b8d3ca] + GR v0.72.8
  [a2bd30eb] + Graphics v1.1.3
  [42e2da0e] + Grisu v1.0.2
  [cd3eb016] + HTTP v1.10.15
  [bbac6d45] + IdentityRanges v0.3.1
  [2803e5a7] + ImageAxes v0.6.9
  [f332f351] + ImageContrastAdjustment v0.3.7
  [a09fc81d] + ImageCore v0.8.22
  [51556ac3] + ImageDistances v0.2.13
  [6a3955dd] + ImageFiltering v0.6.21
  [6218d12a] + ImageMagick v1.3.0
  [bc367c6b] + ImageMetadata v0.9.5
  [787d08f9] + ImageMorphology v0.2.11
  [2996bd0c] + ImageQualityIndexes v0.2.2
  [4e3cecfd] + ImageShow v0.2.3
  [02fcd773] + ImageTransformations v0.8.13
  [916415d5] + Images v0.23.3
  [9b13fd28] + IndirectArrays v0.5.1
  [a98d9a8b] + Interpolations v0.13.6
  [8197267c] + IntervalSets v0.7.10
  [3587e190] + InverseFunctions v0.1.17
  [92d709cd] + IrrationalConstants v0.1.1
  [c8e1da08] + IterTools v1.4.0
  [1019f520] + JLFzf v0.1.9
  [692b3bcd] + JLLWrappers v1.7.0
  [682c06a0] + JSON v0.21.4
  [b964fa9f] + LaTeXStrings v1.4.0
  [23fbe1c1] + Latexify v0.16.6
  [2ab3a3ac] + LogExpFunctions v0.3.28
  [e6f89c97] + LoggingExtras v1.1.0
  [1914dd2f] + MacroTools v0.5.15
  [dbb5928d] + MappedArrays v0.4.2
  [739be429] + MbedTLS v1.1.9
  [442fdcdd] + Measures v0.3.2
  [e1d29d7a] + Missings v1.2.0
  [e94cdb99] + MosaicViews v0.3.4
  [77ba4419] + NaNMath v1.0.3
  [6fe1bfb0] + OffsetArrays v1.15.0
  [4d8831e6] + OpenSSL v1.4.3
  [bac558e1] + OrderedCollections v1.8.0
  [5432bcbf] + PaddedViews v0.5.12
  [d96e819e] + Parameters v0.12.3
  [69de0a69] + Parsers v2.8.1
  [b98c9c47] + Pipe v1.3.0
  [ccf2f8ad] + PlotThemes v3.3.0
  [995b91a9] + PlotUtils v1.4.3
  [91a5bcdd] + Plots v1.40.7
  [7f904dfe] + PlutoUI v0.6.11
  [aea7be01] + PrecompileTools v1.2.1
  [21216c6a] + Preferences v1.4.3
  [94ee1d12] + Quaternions v0.7.6
  [b3c3ace0] + RangeArrays v0.3.2
  [c84ed2f1] + Ratios v0.4.5
  [c1ae055f] + RealDot v0.1.0
  [3cdcf5f2] + RecipesBase v1.3.4
  [01d81517] + RecipesPipeline v0.6.12
  [189a3867] + Reexport v1.2.2
  [05181044] + RelocatableFolders v1.0.1
  [ae029012] + Requires v1.3.1
  [6038ab10] + Rotations v1.7.1
  [6c6a2e73] + Scratch v1.2.1
  [992d4aef] + Showoff v1.0.3
  [777ac1f9] + SimpleBufferStream v1.2.0
  [699a6c99] + SimpleTraits v0.9.4
  [a2af1166] + SortingAlgorithms v1.2.1
  [276daf66] + SpecialFunctions v1.8.8
  [860ef19b] + StableRNGs v1.0.2
  [cae243ae] + StackViews v0.1.1
  [90137ffa] + StaticArrays v1.9.13
  [1e83bf80] + StaticArraysCore v1.4.3
  [82ae8749] + StatsAPI v1.7.0
  [2913bbd2] + StatsBase v0.33.21
  [fd094767] + Suppressor v0.2.8
  [06e1c1a7] + TiledIteration v0.3.1
  [3bb67fe8] + TranscodingStreams v0.11.3
  [5c2747f8] + URIs v1.5.1
  [3a884ed6] + UnPack v1.0.2
  [1cfade01] + UnicodeFun v0.4.1
  [1986cc42] + Unitful v1.22.0
  [45397f5d] + UnitfulLatexify v1.6.4
  [41fe7b60] + Unzip v0.2.0
  [efce3f68] + WoodburyMatrices v0.5.6
  [6e34b625] + Bzip2_jll v1.0.9+0
  [83423d85] + Cairo_jll v1.18.2+1
  [ee1fde0b] + Dbus_jll v1.14.10+0
  [2702e6a9] + EpollShim_jll v0.0.20230411+1
  [2e619515] + Expat_jll v2.6.5+0
  [b22a6f82] + FFMPEG_jll v4.4.2+2
  [f5851436] + FFTW_jll v3.3.10+3
  [a3f928ae] + Fontconfig_jll v2.15.0+0
  [d7e528f0] + FreeType2_jll v2.13.3+1
  [559328eb] + FriBidi_jll v1.0.16+0
  [0656b61e] + GLFW_jll v3.4.0+2
  [d2c73de3] + GR_jll v0.72.8+0
  [78b55507] + Gettext_jll v0.21.0+0
  [7746bdde] + Glib_jll v2.82.4+0
  [3b182d85] + Graphite2_jll v1.3.14+1
  [2e76f6c2] + HarfBuzz_jll v8.5.0+0
  [c73af94c] + ImageMagick_jll v6.9.10-12+3
  [1d5cc7b8] + IntelOpenMP_jll v2025.0.4+0
  [aacddb02] + JpegTurbo_jll v3.1.1+0
  [c1c5ebd0] + LAME_jll v3.100.2+0
  [88015f11] + LERC_jll v3.0.0+1
  [1d63c593] + LLVMOpenMP_jll v18.1.7+0
  [dd4b983a] + LZO_jll v2.10.3+0
  [e9f186c6] + Libffi_jll v3.2.2+2
  [d4300ac3] + Libgcrypt_jll v1.11.0+0
  [7e76a0d4] + Libglvnd_jll v1.7.0+0
  [7add5ba3] + Libgpg_error_jll v1.51.1+0
  [94ce4f54] + Libiconv_jll v1.18.0+0
  [4b2f31a3] + Libmount_jll v2.40.3+0
  [89763e89] + Libtiff_jll v4.4.0+0
  [38a345b3] + Libuuid_jll v2.40.3+0
  [856f044c] + MKL_jll v2025.0.1+1
  [e7412a2a] + Ogg_jll v1.3.5+1
  [458c3c95] + OpenSSL_jll v1.1.23+1
  [efe28fd5] + OpenSpecFun_jll v0.5.6+0
  [91d4177d] + Opus_jll v1.3.3+0
  [36c8627f] + Pango_jll v1.56.1+0
  [30392449] + Pixman_jll v0.43.4+0
  [ea2cea3b] + Qt5Base_jll v5.15.3+2
  [a2964d1f] + Wayland_jll v1.21.0+2
  [2381bf8a] + Wayland_protocols_jll v1.36.0+0
  [02c8fc9c] + XML2_jll v2.13.6+1
  [aed1982a] + XSLT_jll v1.1.42+0
  [4f6342f7] + Xorg_libX11_jll v1.8.6+3
  [0c0b7dd1] + Xorg_libXau_jll v1.0.12+0
  [935fb764] + Xorg_libXcursor_jll v1.2.3+0
  [a3789734] + Xorg_libXdmcp_jll v1.1.5+0
  [1082639a] + Xorg_libXext_jll v1.3.6+3
  [d091e8ba] + Xorg_libXfixes_jll v6.0.0+0
  [a51aa0fd] + Xorg_libXi_jll v1.8.2+0
  [d1454406] + Xorg_libXinerama_jll v1.1.5+0
  [ec84b674] + Xorg_libXrandr_jll v1.5.4+0
  [ea2f1a96] + Xorg_libXrender_jll v0.9.11+1
  [14d82f49] + Xorg_libpthread_stubs_jll v0.1.2+0
  [c7cfdc94] + Xorg_libxcb_jll v1.17.0+3
  [cc61e674] + Xorg_libxkbfile_jll v1.1.2+1
  [12413925] + Xorg_xcb_util_image_jll v0.4.0+1
  [2def613f] + Xorg_xcb_util_jll v0.4.0+1
  [975044d2] + Xorg_xcb_util_keysyms_jll v0.4.0+1
  [0d47668e] + Xorg_xcb_util_renderutil_jll v0.3.9+1
  [c22f9ab0] + Xorg_xcb_util_wm_jll v0.4.1+1
  [35661453] + Xorg_xkbcomp_jll v1.4.6+1
  [33bec58e] + Xorg_xkeyboard_config_jll v2.39.0+0
  [c5fb5394] + Xorg_xtrans_jll v1.5.1+0
  [3161d3a3] + Zstd_jll v1.5.7+1
  [214eeab7] + fzf_jll v0.56.3+0
  [a4ae2306] + libaom_jll v3.11.0+0
  [0ac62f75] + libass_jll v0.15.2+0
  [1183f4f0] + libdecor_jll v0.2.2+0
  [f638f0a6] + libfdk_aac_jll v2.0.3+0
  [b53b4c65] + libpng_jll v1.6.47+0
  [f27f6e37] + libvorbis_jll v1.3.7+2
  [1317d2d5] + oneTBB_jll v2022.0.0+0
  [1270edf5] + x264_jll v2021.5.5+0
  [dfaa095f] + x265_jll v3.5.0+0
  [d8fb68d0] + xkbcommon_jll v1.4.1+2
  [0dad84c5] + ArgTools
  [56f22d72] + Artifacts
  [2a0f44e3] + Base64
  [ade2ca70] + Dates
  [8bb1440f] + DelimitedFiles
  [8ba89e20] + Distributed
  [f43a241f] + Downloads
  [7b1f6079] + FileWatching
  [b77e0a4c] + InteractiveUtils
  [4af54fe1] + LazyArtifacts
  [b27032c2] + LibCURL
  [76f85450] + LibGit2
  [8f399da3] + Libdl
  [37e2e46d] + LinearAlgebra
  [56ddb016] + Logging
  [d6f4376e] + Markdown
  [a63ad114] + Mmap
  [ca575930] + NetworkOptions
  [44cfe95a] + Pkg
  [de0858da] + Printf
  [3fa0cd96] + REPL
  [9a3f8284] + Random
  [ea8e919c] + SHA
  [9e88b42a] + Serialization
  [1a1011a3] + SharedArrays
  [6462fe0b] + Sockets
  [2f01184e] + SparseArrays
  [10745b16] + Statistics
  [fa267f1f] + TOML
  [a4e569a6] + Tar
  [8dfed614] + Test
  [cf7118a7] + UUIDs
  [4ec0a83e] + Unicode
  [e66e0078] + CompilerSupportLibraries_jll
  [deac9b47] + LibCURL_jll
  [29816b5a] + LibSSH2_jll
  [c8ffd9c3] + MbedTLS_jll
  [14a3606d] + MozillaCACerts_jll
  [4536629a] + OpenBLAS_jll
  [05823500] + OpenLibm_jll
  [efcefdf7] + PCRE2_jll
  [83775a58] + Zlib_jll
  [8e850b90] + libblastrampoline_jll
  [8e850ede] + nghttp2_jll
  [3f19e933] + p7zip_jll
9.9 s

From the last homework

Below we have included some important functions from the last homework (Raytracing in 2D), which we will be able to re-use for the 3D case.

There are some small changes:

  1. The concept of a Photon now carries color information.

  2. The Sphere is no longer a pure lens, it contains a Surface property which describes the mixture between transmission, reflection and a pure color. More on this later!

  3. The refract function is updated to handle two edge cases, but its behaviour is generally unchanged.

Outside of these changes, all functions from the previous homework can be taken "as-is" when converting to 3D, cool!

👀 Reading hidden code
6.0 ms
struct Photon
"Position vector"
p::Vector{Float64}

"Direction vector"
l::Vector{Float64}

"Color associated with the photon"
c::RGB
ior::Real
end
👀 Reading hidden code
1.9 ms

Intersections:

👀 Reading hidden code
5.0 ms
abstract type Object end
👀 Reading hidden code
405 μs
struct Miss end
👀 Reading hidden code
873 μs
struct Intersection{T<:Object}
object::T
distance::Float64
point::Vector{Float64}
end
👀 Reading hidden code
4.0 ms
begin
Base.isless(a::Miss, b::Miss) = false
Base.isless(a::Miss, b::Intersection) = false
Base.isless(a::Intersection, b::Miss) = true
Base.isless(a::Intersection, b::Intersection) = a.distance < b.distance
end
👀 Reading hidden code
1.3 ms
closest_hit (generic function with 1 method)
function closest_hit(photon::Photon, objects::Vector{<:Object})
hits = intersection.([photon], objects)
minimum(hits)
end
👀 Reading hidden code
654 μs

Reflect and refract:

👀 Reading hidden code
196 μs
reflect (generic function with 1 method)
reflect(ℓ₁::Vector, n̂::Vector)::Vector = normalize(ℓ₁ - 2 * dot(ℓ₁, n̂) * n̂)
👀 Reading hidden code
583 μs
refract (generic function with 1 method)
function refract(
ℓ₁::Vector, n̂::Vector,
old_ior, new_ior
)
r = old_ior / new_ior
n̂_oriented = if -dot(ℓ₁, n̂) < 0
-n̂
else
end
c = -dot(ℓ₁, n̂_oriented)
if abs(c) > 0.999
ℓ₁
else
f = 1 - r^2 * (1 - c^2)
if f < 0
ℓ₁
else
normalize(r * ℓ₁ + (r*c - sqrt(f)) * n̂_oriented)
end
end
end
👀 Reading hidden code
1.2 ms

Surface (new)

👀 Reading hidden code
194 μs
struct Surface
# Reflectivity
r::Float64

# Transmission
t::Float64

# Color
c::RGBA

# index of refraction
ior::Float64

end
👀 Reading hidden code
1.9 ms

Sphere

Aasdf

html"""
<h4 id="sphere-defs">Sphere</h4>
<p>Aasdf</p>
"""
👀 Reading hidden code
124 μs
struct Sphere <: Object
# Lens position
p::Vector{Float64}

# Lens radius
r::Float64

s::Surface
end
👀 Reading hidden code
1.5 ms
intersection (generic function with 1 method)
function intersection(photon::Photon, sphere::S; ϵ=1e-3) where {S <: Union{SkyBox, Sphere}}
a = dot(photon.l, photon.l)
b = 2 * dot(photon.l, photon.p - sphere.p)
c = dot(photon.p - sphere.p, photon.p - sphere.p) - sphere.r^2
d = b^2 - 4*a*c
if d <= 0
Miss()
else
t1 = (-b-sqrt(d))/2a
t2 = (-b+sqrt(d))/2a
t = if t1 > ϵ
t1
elseif t2 > ϵ
t2
else
return Miss()
end
point = photon.p + t * photon.l
Intersection(sphere, t, point)
end
end
👀 Reading hidden code
2.7 ms
sphere_normal_at (generic function with 1 method)
function sphere_normal_at(p::Vector{Float64}, s::Sphere)
normalize(p - s.p)
end
👀 Reading hidden code
457 μs

Camera and Skyboxes

Now we can begin looking into the 3D nature of raytracing to create visualizations similar to those in lecture. The first step is setting up the camera and another stuct known as a sky box to collect all the rays of light.

Luckily, the transition from 2D to 3D for raytracing is relatively straightforward and we can use all of the functions and concepts we have built in 2D moving forward.

Firstly, the camera:

👀 Reading hidden code
7.5 ms

Camera

For the purposes of this homework, we will constrain ourselves to a camera pointing exclusively downward. This is simply because camera positioning can be a bit tricky and there is no reason to make the homework more complicated than it needs to be!

So, what is the purpose of the camera?

Well, in reality, a camera is a device that collects the color information from all the rays of light that are refracting and reflecting off of various objects in some sort of scene. Because there are a nearly infinite number of rays bouncing around the scene at any time, we will actually constrain ourselves only to rays that are entering our camera. In poarticular, we will create a 2D screen just in front of the camera and send a ray from the camera to each pixel in the screen, as shown in the following image:

👀 Reading hidden code
33.7 ms

Because we are not considering camera motion for this exercise, we will assume that the image plane is constrained to the horizontal plane, but that the camera, itself, can be some distance behind it. This distance from the image plane to the camera is called the focal length and is used to determine the field of view.

From here, it's clear we need to construct:

  1. A camera struct

  2. A function to initialize all the rays being generated by the camera

Let's start with the struct

👀 Reading hidden code
518 μs
struct Camera <: Object
"Set of all pixels, counts as scene resolution"
resolution::Tuple{Int64,Int64}

"Physical size of aperture"
aperture_width::Float64

"Camera's distance from screen"
focal_length::Float64

"Camera's position"
p::Vector{Float64}
end
👀 Reading hidden code
1.7 ms
test_cam = Camera((400,300), 9, -10, [0,00,100])
👀 Reading hidden code
19.2 μs

Now we need to construct some method to create each individual ray extending from the camera to a pixel in the image plane.

👀 Reading hidden code
182 μs
init_rays (generic function with 1 method)
function init_rays(cam::Camera)
# Physical size of the aperture/image/grid
aspect_ratio = cam.resolution[1] / cam.resolution[2]
dim = (
cam.aperture_width,
cam.aperture_width / aspect_ratio
)

# The x, y coordinates of every pixel in our image grid
# relative to the image center
xs = LinRange(-0.5* dim[1], 0.5 * dim[1], cam.resolution[1])
ys = LinRange(0.5* dim[2], -0.5 * dim[2], cam.resolution[2])
pixel_positions = [[x, y, cam.focal_length] for y in ys, x in xs]
directions = normalize.(pixel_positions)
Photon.([cam.p], directions, [zero(RGB)], [1.0])
end
👀 Reading hidden code
2.2 ms
tiny_resolution_camera = Camera((4,3), 16, -5, [0, 20, 100])
👀 Reading hidden code
19.9 μs
3×4 Matrix{Photon}:
 Photon([0.0, 20.0, 100.0], [-0.715542, 0.536656, -0.447214], RGB{N0f8}(0.0,0.0,0.0), 1.0)   …  Photon([0.0, 20.0, 100.0], [0.715542, 0.536656, -0.447214], RGB{N0f8}(0.0,0.0,0.0), 1.0)
 Photon([0.0, 20.0, 100.0], [-0.847998, 0.0, -0.529999], RGB{N0f8}(0.0,0.0,0.0), 1.0)           Photon([0.0, 20.0, 100.0], [0.847998, 0.0, -0.529999], RGB{N0f8}(0.0,0.0,0.0), 1.0)
 Photon([0.0, 20.0, 100.0], [-0.715542, -0.536656, -0.447214], RGB{N0f8}(0.0,0.0,0.0), 1.0)     Photon([0.0, 20.0, 100.0], [0.715542, -0.536656, -0.447214], RGB{N0f8}(0.0,0.0,0.0), 1.0)
test_rays = init_rays(tiny_resolution_camera)
👀 Reading hidden code
23.9 μs
extract_colors (generic function with 1 method)
extract_colors(rays) = map(ray -> ray.c, rays)
👀 Reading hidden code
886 μs
extract_colors(test_rays)
👀 Reading hidden code
26.5 ms

Nothing yet... time to add some objects!

Skybox

Now that we have the concept of a camera, we can technically do a fully 3D raytracing example; however, we want to ensure that each pixel will actually hit something – preferrably something with some color gradient so we can make sure our simulation is working!

For this, we will introduce the concept of a sky box, which is standard for most gaming applications. Here, the idea is that our entire scene is held within some additional object, just like the mirrors we used in the 2D example. The only difference here is that we will be using some texture instead of a reflective surface. In addition, even though we are calling it a box, we'll actually be treating it as a sphere.

Because we have already worked out how to make sure we have hit the interior of a spherical lens, we will be using a similar function here. For this part of the exercise, we will need to construct 2 things:

  1. A skybox struct

  2. A function that returns some color gradient to be called whenever a ray of light interacts with a sky box

So let's start with the sky box struct

👀 Reading hidden code
6.1 ms
struct SkyBox <: Object
# Skybox position
p::Vector{Float64}

# Skybox radius
r::Float64
# Color function
c::Function
end
👀 Reading hidden code
1.5 ms

Now we have the ability to create a skybox, the only thing left is to create some sort of texture function so that when the ray of light hits the sky box, we can return some form of color information. So for this, we will basically create a function that returns back a smooth gradient in different directions depending on the position of the ray when it hits the skybox.

For the color information, we will be assigning a color to each cardinal axis. That is to say that there will be a red gradient along x, a blue gradient along y, and a green gradient along z. For this, we will need to define some extent over which the gradient will be active in 'real' units. From there, we can say that the gradient is

r+D2D,

where r is the ray's position when it hits the skybox, and D is the extent over which the gradient is active.

So let's get to it and write the function!

👀 Reading hidden code
5.1 ms
gradient_skybox_color (generic function with 1 method)
function gradient_skybox_color(position, skybox)
extents = skybox.r
c = zero(RGB)
if position[1] < extents && position[1] > -extents
c += RGB((position[1]+extents)/(2.0*extents), 0, 0)
end

if position[2] < extents && position[2] > -extents
c += RGB(0,0,(position[2]+extents)/(2.0*extents))
end

if position[3] < extents && position[3] > -extents
c += RGB(0,(position[3]+extents)/(2.0*extents), 0)
end

return c
end
👀 Reading hidden code
1.4 ms
sky = SkyBox([0.0, 0.0, 0.0], 1000, gradient_skybox_color)
👀 Reading hidden code
17.8 μs

Let's set up a basic scene and trace an image! Since our skybox is spherical we can use the same intersect method as we use for Spheres. Have a look at the intersect method, we already added SkyBox as a possible type.

👀 Reading hidden code
439 μs
basic_camera = Camera((300,200), 16, -5, [0,20,100])
👀 Reading hidden code
19.7 μs
let
scene = [sky]
ray_trace(scene, basic_camera; num_intersections=4)
end
👀 Reading hidden code
2.0 s

To create this image, we used the ray tracing function bewlow, which takes in a camera and a set of objects / scene, and...

  1. Initilializes all the rays

  2. Propagates the rays forward

  3. Converts everything into an image

👀 Reading hidden code
438 μs
ray_trace (generic function with 1 method)
function ray_trace(objects::Vector{O}, cam::Camera;
num_intersections = 10) where {O <: Object}
rays = init_rays(cam)

new_rays = step_ray.(rays, [objects], [num_intersections])

extract_colors(new_rays)
end
👀 Reading hidden code
1.7 ms

Writing a ray tracer

It's your turn! Below is the code needed to trace just the sky box, but we still need to add the ability to trace spheres.

We recommend that you start by just implementing reflection - make every sphere reflect, regardless of its surface. Make sure that this is working well - can you see the reflection of one sphere in another sphere? Does our program get stuck in a loop?

Once you have reflections working, you can add refraction and colored spheres. In the 2D example, we dealt specifically with spheres that could either 100% reflect or refract. In reality, it is possible for objects to either reflect or refract, something in-between. That is to say, a ray of light can split when hitting an object surface, creating at least 2 more rays of light that will both return separate color values. The color that we perceive for that ray is the combination both of these colors - they are mixed.

A third possibility explored in the lecture is that the objects can also have a color associated with them and just return the color value instead of reflecting or refracting.

You can choose! After implementing reflection, you can implement three different spheres (you can modify the existing code, create new types, add functions, and so on), a purely reflective, purely refractive or opaquely colored sphere. You can also go straight for the more photorealistic option, which is that every sphere is a combination of these three - this is what we did in the lecture.

👀 Reading hidden code
774 μs
interact (generic function with 1 method)
function interact(ray::Photon, hit::Intersection{SkyBox}, ::Any, ::Any)
ray_color = hit.object.c(hit.point, hit.object)
Photon(hit.point, ray.l, ray_color, ray.ior)
end
👀 Reading hidden code
709 μs
interact (generic function with 2 methods)
interact(photon::Photon, ::Miss, ::Any, ::Any) = photon
👀 Reading hidden code
386 μs
step_ray (generic function with 1 method)
begin
function interact(ray::Photon, hit::Intersection{Sphere}, num_intersections, objects)
sphere = hit.object

reflected_ray = ray
refracted_ray = ray
colored_ray = ray

if !isapprox(sphere.s.t, 0)
old_ior = ray.ior
new_ior = if ray.ior == 1.0
sphere.s.ior
else
1.0
end

normal = sphere_normal_at(hit.point, hit.object)

refracted_ray = Photon(hit.point, refract(ray.l, normal, old_ior, new_ior), ray.c, new_ior)
refracted_ray = step_ray(refracted_ray, objects,
num_intersections-1)
end

if !isapprox(sphere.s.r, 0)
n = sphere_normal_at(hit.point, sphere)
reflected_ray = Photon(hit.point, reflect(ray.l, n), ray.c, ray.ior)
reflected_ray = step_ray(reflected_ray, objects,
num_intersections-1)
end

if !isapprox(sphere.s.c.alpha, 0)
ray_color = RGB(sphere.s.c)
colored_ray = Photon(ray.l, ray.p, ray_color, ray.ior)
end

ray_color = sphere.s.t * refracted_ray.c +
sphere.s.r * reflected_ray.c +
sphere.s.c.alpha*colored_ray.c

Photon(hit.point, ray.l, ray_color, ray.ior)
end
function step_ray(ray::Photon, objects::Vector{O},
num_intersections) where {O <: Object}

if num_intersections == 0
ray
else
hit = closest_hit(ray, objects)
interact(ray, hit, num_intersections, objects)
end
end
end
👀 Reading hidden code
2.9 ms
# function step_ray(ray::Photon, objects::Vector{O},
# num_intersections) where {O <: Object}

# if num_intersections == 0
# ray
# else
# hit = closest_hit(ray, objects)
# interact(ray, hit, num_intersections, objects)
# end
# end
👀 Reading hidden code
14.1 μs
let
cam = Camera((600,360), 16, -15, [0,10,100])

ray_trace(main_scene, cam; num_intersections=3)
end
👀 Reading hidden code
2.6 s

Below, we create a scene with a number of balls inside of it. While working on your code, work in small increments, and do frequent checks to see if your code is working. Feel free to modify this test scene, or to create a simpler one.

👀 Reading hidden code
216 μs
main_scene = [
sky,
Sphere([0,0,-25], 20,
Surface(1.0, 0.0, RGBA(1,1,1,0.0), 1.5)),
Sphere([0,50,-100], 20,
Surface(0.0, 1.0, RGBA(0,0,0,0.0), 1.0)),
Sphere([-50,0,-25], 20,
Surface(0.0, 0.0, RGBA(0, .3, .8, 1), 1.0)),
Sphere([30, 25, -60], 20,
Surface(0.0, 0.75, RGBA(1,0,0,0.25), 1.5)),
Sphere([50, 0, -25], 20,
Surface(0.5, 0.0, RGBA(.1,.9,.1,0.5), 1.5)),
Sphere([-30, 25, -60], 20,
Surface(0.5, 0.5, RGBA(1,1,1,0), 1.5)),
]
👀 Reading hidden code
31.3 ms

Hint

If you are getting a "Circular Defintions" error - this could be because of a Pluto limitation. If two functions call each other, they need to be contained in a single cell, using a begin end block.

👀 Reading hidden code
45.7 ms

Bonus: Escher

If you managed to get through the exercises, we have a fun bonus exercise! The goal is to recreate this self-portrait by M.C. Escher:

👀 Reading hidden code
261 μs
👀 Reading hidden code
4.5 ms

It looks like M.C. Escher is a skillful raytracer, but so are we! To recreate this image, we can simplify it by having just two objects in our scene:

  • A purely reflective sphere

  • A skybox, containing an image of us!

Let's start with our old skybox, and set up our scene:

👀 Reading hidden code
410 μs
escher_sphere = Sphere([0,0,0], 20,
Surface(1.0, 0.0, RGBA(1,1,1,0.0), 1.5))
👀 Reading hidden code
22.6 μs
escher_cam = Camera((300,300), 30, -10, [0,00,30])
👀 Reading hidden code
20.0 μs

👆 You can modify escher_cam to increase or descrease the resolution!

👀 Reading hidden code
190 μs
let
scene = [sky, escher_sphere]
ray_trace(scene, escher_cam; num_intersections=3)
end
👀 Reading hidden code
378 ms

Awesome! Next, we want to set an image as our skybox, instead of a gradient. To do so, we have written a function that converts the x,y,z coordinates of the intersection point with a skybox into a latitude/longitude pair, which we can use (after scaling, rounding & clamping) as pixel coordinates to index an image!

👀 Reading hidden code
243 μs
👀 Reading hidden code
1.7 s
image_skybox (generic function with 1 method)
function image_skybox(img)
f = function(position, skybox)
lon = atan(-position[1], position[3])
lat = -atan(position[2], norm(position[[1,3]]))

get_index_rational(img, (lat/(pi)) + .5, (lon/2pi) + .5)
end
SkyBox([0.0, 0.0, 0.0], 1000, f)
end
👀 Reading hidden code
1.4 ms
get_index_rational (generic function with 1 method)
function get_index_rational(A, x, y)
a, b = size(A)
u = clamp(floor(Int, x * (a-1)) + 1, 1, a)
v = clamp(floor(Int, y * (b-1)) + 1, 1, b)
A[u,v]
end
👀 Reading hidden code
984 μs
earth_skybox = image_skybox(earth)
👀 Reading hidden code
15.2 μs
let
scene = [earth_skybox, escher_sphere]
ray_trace(scene, escher_cam; num_intersections=3)
end
👀 Reading hidden code
465 ms

Great! It's like the Earth, but distorted. Notice that the continents are mirrored, and that you can see both poles at the same time.

Okay, self portrait time! Let's take a picture using your webcam, and we will use it as the skybox texture:

👀 Reading hidden code
232 μs
Enable webcam
@bind wow camera_input()
👀 Reading hidden code
241 ms
Error message

MethodError: no method matching getindex(::Missing, ::String)

Stack trace

Here is what happened, the most recent locations are first:

  1. process_raw_camera_data(raw_camera_data::Missing)
    	# So to get the red values for each pixel, we take every 4th value, starting at 	# the 1st:	reds_flat = UInt8.(raw_camera_data["data"][1:4:end])	greens_flat = UInt8.(raw_camera_data["data"][2:4:end])	blues_flat = UInt8.(raw_camera_data["data"][3:4:end])
  2. Show more...
beep boop CRASH 🤖
face = process_raw_camera_data(wow)
👀 Reading hidden code
---
Error message

Another cell defining face contains errors.

Everything is going to be okay!
let
face_skybox = image_skybox(face)
scene = [face_skybox, escher_sphere]
ray_trace(scene, escher_cam; num_intersections=3)
end
👀 Reading hidden code
---

👀 wow! It's Planet Jazzy Doe, surrounded by even more Jazzy Doe.

When you look at the drawing by Escher, you see that he only occupies a small section of the 'skybox'. Behind Escher, you can see his cozy house, and in front of him (i.e. behind the glass sphere, from his persective), you see a gray background.

What we need is a 360° panoramic image of our room. One option is that you make one! There are great tutorials online, and maybe you can use an app to do this with your phone.

Another option is that we approximate the panaroma by padding the image of your face. You can pad the image with a solid color, you can use a gradient, you can use the earth satellite image, you can add noise, and so on. Here is an example of what I made.

👀 Reading hidden code
860 μs
padded (generic function with 1 method)
function padded(face)
a,b = size(face)
Abw = [((2a-y) / 2a) for y in 1:2a, x in 1:3b]
Abw .-= .1 * rand(Gray, 2a, 3b)
A = RGB.(Abw)
c = a ÷ 2
d = b ÷ 2
A[
c + 1:c + a,
b + 1:2b] .= face
A
end
👀 Reading hidden code
2.2 ms
Error message

Another cell defining face contains errors.

Probably not your fault!
padded(face)
👀 Reading hidden code
---

Let's put it all together!

👀 Reading hidden code
178 μs
Enable webcam
👀 Reading hidden code
724 μs
Error message

MethodError: no method matching getindex(::Missing, ::String)

Stack trace

Here is what happened, the most recent locations are first:

  1. process_raw_camera_data(raw_camera_data::Missing)
    	# So to get the red values for each pixel, we take every 4th value, starting at 	# the 1st:	reds_flat = UInt8.(raw_camera_data["data"][1:4:end])	greens_flat = UInt8.(raw_camera_data["data"][2:4:end])	blues_flat = UInt8.(raw_camera_data["data"][3:4:end])
  2. Show more...
let
img = process_raw_camera_data(escher_face_data)
img_padded = padded(img)
scene = [
image_skybox(padded(img)),

escher_sphere,
]
ray_trace(scene, escher_cam; num_intersections=20)
end
👀 Reading hidden code
---

Before you submit

Remember to fill in your name and Kerberos ID at the top of this notebook.

👀 Reading hidden code
473 μs

Function library

Just some helper functions used in the notebook.

👀 Reading hidden code
216 μs
process_raw_camera_data (generic function with 1 method)
👀 Reading hidden code
1.5 ms
camera_input (generic function with 1 method)
👀 Reading hidden code
1.5 ms
hint (generic function with 1 method)
👀 Reading hidden code
493 μs
almost (generic function with 1 method)
👀 Reading hidden code
490 μs
still_missing (generic function with 2 methods)
👀 Reading hidden code
824 μs
keep_working (generic function with 2 methods)
👀 Reading hidden code
873 μs
👀 Reading hidden code
11.5 ms
correct (generic function with 2 methods)
correct(text=rand(yays)) = Markdown.MD(Markdown.Admonition("correct", "Got it!", [text]))
👀 Reading hidden code
749 μs
not_defined (generic function with 1 method)
👀 Reading hidden code
810 μs
TODO
👀 Reading hidden code
33.3 μs
TODO_note (generic function with 1 method)
TODO_note(text) = Markdown.MD(Markdown.Admonition("warning", "TODO note", [text]))
👀 Reading hidden code
466 μs